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^<device> 

<iconList> 
<icon> 
<size>16</size> 
<color>0</colop> 
<depth>8</depth> 
<imageType>PNG</imageType> 

<image>"http://device.local/iconpath/icon16bw.png"</image> 
</icon> 
<icon> 

<size>32</size> 

<color>0</color> 

<depth>8</depth> 

<imageType>PNG</imageType> 

<image>"http://deviceJocal/iconpath/icon32bw.png"</image> 
</icon> 
<icon> 

<size>48</size> 

<CX)IOP^0</COlOP> 

<depth>8</depth> 
<imageType>PNG</imageType> 

<image>"http://device.local/iconpath/icon48bw.png"</image> 

</icx»n> 
<icon> 

<size>16</size> 

<color>1</coloP> 

<depth>8</depth> 

<imageType>PNG</imageType> 

<image>"http://device.local/iconpath/icon16c.png"</image> 

</icon><device> 
<icon> 

<size>32</size> 

<color>0</color> 

<depth>8</depth> 

<imageType>PNG</imageType> 

<image>"http://device.local/iconpath/icon32c.png"</image> 
</icon> 
<icon> 

<size>48</size> 

<color>0</color> 

<depth>8</depth> 

<imageType>PNG</imageType> 

<image>"http://device.local/iconpath/icon48c.png"</innage> 

</icon> 
</iconList> 
^</device> 
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<?xml version="1.0"?> 
<scpd xmlns="x-schema:scpdl-schema.xnnr 
<service StateTable> 
<stateVariable> 
<name>currentChannel</name> 
<dataType>nunnber</dataType> 
<allowedValueRange> 
<minimum>0</minimum> 
<maximum>55</maximunn> 
<step>1</step> 
</allowedValueRange> 
</stateVariable> 
</serviceStateTable> 



<actionList> 
<action> 

<name>ChannelUp</name> 
</action> 

<action> 

<name>ChannelDown</name> 
</action> 

<action> 

<name>SetChannel</name> 
<argument> 
<name>newChannel</name> 
<relatedStateVariable> 
currentChannel 
</relatedStateVariable> 
</argunnent> 
</action> 
</actionList> 
</scpd> 
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<contract> 
<protocol id="protocolDef' > 
<HTTP version="1.1"> 
<URL></URL> 
<M-POST> 

<MAN>http://www.microsoft.corn/protocols/ext/XOAP</MAN> 
</M-POST> 

<HEADER name="Content-Type" value="text/xmr /> 
<!- Need to put in extension headers liere -> 
</HTTP> 



</protocol> 

<RequestResponse name="queryStateVariable"> 
<protocol is="protocolDef* > 
<in is="queryStateVariable"> 
<out is="queryStateVariableResponse"> 
<error is="queryStateVariableResponse"> 

</RequestResponse> 

<RequestResponse name="invokeAction"> 

<protocol is="protocolDef*> 

<in is="SerializedStream"> 

<out is="invokeActionResponse"> 

<error is="invokeActionResponse"> 
</RequestResponse> 



<Schema nanne="upnp_scpdl" 
xmlns-'urn:schemas-microsoft-conn:xml-data" 
xmlns:dt="urn:schemas-microsofl-com:datatypes"> 

<!~ Common -> 

<ElementType name="_return" content="textOnly" dt:type="string" /> 
<ElementType name="Jauir content="textOnly" dt:type="string" /> 

<!- Query State Variable Call -> 

<ElementType name='VariableName" content="textOnly" dt:type="string" /> 

<ElementType name="queryStateVariable" content="eltOnly" model="closed" 

<element type-VariableName" /> 
</ElementType> 

<!- Query State Variable Response -> 

\ 
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<ElementType name="queryStateVariableResponse" content="eltOnly" 
model="closed"> 

<group order="one"> 
<element type="_return"> 
<element type="_fault"> 
</group> 
</ElementType> 

<!- Invoke Action Call -> 

<AttributeType name="main" dt:type="idref' /> 
<AttributeType name="headers" dt:type="idref ' /> 
<AttributeType name="id" dt:type="id" /> 

<ElementType name="sequenceNunfiber" content="textOnly" dt:type="int"> 
<AttrbuteType name="dt" dt:type="string" dt:values="int" /> 

<attribute type="dt" /> 
</ElementType> 

<ElementType name="headers" content="eltOnly" model="closed" 

<attribute type="id" required-'yes" /> 

<element type="sequenceNumber" /> 
</ElementType> 

<ElementType name="actionName" content="textOnly" dt:type="string" /> 
<ElementType name="actionArg" content="textOnly" dt:type="string" /> 

<ElementType name="invokeAction" content="eltOnly" model="closed"> 
<attribute type="id" required="yes" /> 

<element type="actionName"> 

<element type-'actionArg" minOccurs- "O" maxOccurs="*"> 
</ElementType> 
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<ElementType name="SerializedStream" content="eltOnly" model="closed"> 
<attribute type="main" required="yes" /> 
ottribute type="headers" required="yes" /> 

<element type="headers"> 
<element type="invokeAction"> 

</ElementType> 

<!- Invoke Action Response ~> 

<ElementType name="invokeActionResponse" content="eltOnly" model="closed"> 
<group ordep="one"> 
<element type="_return"> 
<element type="_fault"> 
</group> 
</ElementType> 
</Schema> 
</contract> 
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<?xml version="1.0"?> 
<Schema name- 'upnp_scpcH" 

xmlns="urn:schemas-microsoft-com:xml-data" 

xmlns:dt="urn:schemas-microsoft-com:datatypes"> 

<!- Common Elements and Attributes ~> 

<ElementType name="name" content="textOnly" dt:type="string" /> 
<!- Service State Table ~> 

<ElementType name="minimum" content- 'textOnly" dt:type="number" /> 
<ElementType name="maximum" content-'textOnly" dt:type="number" /> 
<ElementType name="step" content-'textOnly" dt:type="number" /> 

<ElementType name="allowedValueRange" content-'eltOnly" model="closed"> 

<element type-'minimum" /> 

<eiement type="maximum" /> 

<element type="step" minOccurs="0" /> 
</ElementType> 

<ElementType name="allowedValue" content="textOnly" /> 

<ElementType name="allowedValueList" content="eltOnly" model="closed"> 

<element type="ailowedValue" minOccurs-'1" maxOccurs="*" /> 
</ElementType> 

<ElementType name="dataType" content-'textOnly" dt:type="string" /> 

<ElementType name="stateVariable" content="eltOnly" model="closed"> 
<element type="name" /> 
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<element type="dataType" /> 

<group minOccurs="0" maxOccurs="1" order="one"> 
<element type="allowedValueRange" /> 
<element type="allowedValueList" /> 

</group> 
</ElementType> 



<ElementType name="deviceStateTable" content="eltOnly" model="closed"> 

<element type="stateVariable" nninOccurs="1" maxOccurs="*" /> 
</ElementType> 



<!~ Action List 

<ElennentType name="relatedStateVariable" content="textOnly" dt:type="string" /> 

<ElennentType name="argunrient" content="eltOnly" model="closed"> 

<element type="name" /> 

<element type="relatedStateVariable" /> 
</ElementType> 

<ElementType name="action" content="eltOnly" model="closed"> 
<element type="name" /> 

<element type="argument" minOccurs="0" maxOccurs="*" /> 
</ElennentType> 

<ElementType name="actionList" content="eltOnly" model="closed"> 

<element type="action" minOccurs="0" maxOccurs="*" /> 
</ElementType> 

<!- Root Element --> 

<ElementType name-'dcpd" content="eltOnly" model="closed"> 

<element type="deviceStateTable" /> 

<element type="actionList" /> 
</ElementType> 
</Schenna> 
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I 

object, 

uuid(<foo>), 

dual, 

helpstringflUPNPDevice interface"), 
pointer_default(unique) 

1 

interface lUPNPDevice : IDispatch 

{ 

[propget. id(DISPID_UPNPDEVICE_DESCRIPTIONDOCUMENT), 
helpstringC'returns the document from which the properties of this device are 
being read")] 

HRESULT DescriptionDocument([restricted, hidden, out, retval] 
lUPNPDescriptionDocument ** ppuddDocument); 

purpose: retums the document from which the properties of this device are 
being read. 

parameters: ppuddDocument, A reference to the description document 
object from which data about the device is being read. This must be freed when no 
longer needed. 

return values: S_OK, ppuddDocument is a refernce to the device's 
description document. 

[propget, id(DISPID_UPNPDEVICE_ISROOTDEVICE), 
helpstring("denotes whether the physical location information of this device can 
be set")] 

HRESULT lsRootDevice([out, retval] VARIANT_BOOL * pvarb); 

parameters: pvarb, the address of a VARIANT_BOOL that will receive the 
value of VARIANT_TRUE if the current device is the topmost device in the device 
tree, and will receive the value of VARIANT_FALSE othenwise. 
return values: S_OK, varb is set to the appropriate value 
note: if a device is a root device, calls RootDeviceQ or ParentDeviceQ will 
return NULL 



[propget, id(DISPID_UPNPDEVICE_ROOT), 
helpstring("returns the top device in the device tree")] 
HRESULT RootDevice([out, retval] lUPNPDevice ** ppudDeviceRoot); 
purpose: returns the top device in the device tree 



\ 
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parameters: ppudDeviceRoot, On return, this refers to the "root" device of 
the current device tree. The root device is the topmost parent of the current device. 
If the current device is the root device this method will set *ppudDeviceRoot to null, 
and return S_FALSE. 

return values: S_OK, *ppudDeviceRoot contains a reference to the root 
device. S_FALSE, the current device is the root device. *ppudDeviceRoot is null. 

[propget, id(DISPID_UPNPDEVICE_PARENT), 
helpstringC'returns the parent of the current device")] 
HRESULT ParentDevice([out, retval] lUPNPDevice ** ppudDeviceParent); 

parameters: ppudDeviceParent, On return, if the device has a parent, this Is 

the address of a lUPNPDevice object which can describe the parent. This must be 

released when no longer needed. If the device has no parent (it is a "root" device), 

than this value will be set to null. 

return values: S_OK, ppudDeviceParent contains a reference to the device's 

parent. S_FALSE, the current device is the root device, which has no parent. 

*ppudDeviceRoot is null. 

[propget, id(DISPID_UPNPDEVICE_CHILDREN), 
helpstringC'returns a collection of the children of the current device")] 
HRESULT Children([out, retval] lUPNPDevices ** ppudChildren); 

parameters: ppudChildren, On retum, this is the address of a newly-created 
lUPNPDevices collection that can enumerate this device's children. This must be 
released when no longer needed. If the device has no children, this method will 
return a collection object with a length of zero. 

return values: S_OK, ppudChildren contains a list of the device's children. 

[propget, id(DISPID_UPNPDEVICE_UDN), 
helpstringC'returns the UDN of the device")] 
HRESULT UniqueDeviceName([out, retval] BSTR * pbstrUDN); 

parameters: pbstrUDN, On return, this contains the address of a newly- 
allocated string which contains the device's Unique Device Name (UDN). The UDN 
is globally unique across all devices - no two devices will ever have the same UDN. 
This value must be freed when no longer needed. 

return values: S_OK pbstrUDN contains the UDN of the device 
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[propget, id(DISPID_UPNPDEVICE_DISPLAYNAME), 
helpstringC'returns the (optional) display name of the device")] 
HRESULT DisplayName([out, retval] BSTR * pbstrDlspiayName); 

parameters: pbstrDisplayName, On return, this contains the address of the 
device's display name. This value must be freed when no longer needed. If the 
device does not specify a display name, this parameter will be set to null. 

return values: S_OK, bstrDisplayName contains the display name of the 
device. pbstrDisplayName must be freed. S_FALSE, the device did not specify a 
display name. ^pbstrDisplayName is set to null. 

note: it is possible for multiple devices to have the same display name. 
Applications should use UniqueDeviceNameQ to determine if two device objects 
refer to the same device. 

[propget. id(DISPID_UPNPDEVICE_CANSETDISPLAYNAME), 
helpstrlngC'denotes whether the physical location information of this device can 
be set")] 

HRESULT CanSetDisplayName([out, retval] VARIANT_BOOL * pvarb); 

parameters: pvarb, the address of a VARIANT_BOOL. This is true (1=0) on 
return when the device's display name can be set (via SetDisplayName) 
return values: S_OK varb Is set to the appropriate value 

[id(DISPID_UPNPDEVICE_SETDISPLAYNAME), 
helpstringC'sets the display name on the device")] 
HRESULT SetDisplayName([in] BSTR bstrDisplayName); 

parameters: bstrDisplayName, the value to set the device's display name to. 
return values: S_OK, varb is set to the appropriate value, 
note: On success, this method sets the display name used by a device. 
Note that this method changes the display name on the device itself, not simply on 
the local object. This will block while the name is being set. 
Additionally, this change will be made on the device alone, and will not be reflected 
in the current device object. After a successful call to this method, DisplayName 
will continue to return the 'old' value). To read the device's current name, the caller 
must re-load the device's description. 



[propget, id(DISPID_UPNPDEVICE_DEVICETYPE), 
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helpstringC'returns the device type URI")] 

HRESULT Type([out, retvalj BSTR * pbstrType); 

, parameters: pbstrType, On return, this contains the address of a newly-allocated 
string containing the device's type URI. This value must be freed when no longer 
needed. 

return values: S_OK, bstrType contains the type URI of the device, and must be 
freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICE_SERVICES). 
helpstringC'returns the collection of services exposed by the device")! 
HRESULT Services([out, retval] lUPNPServices ** ppusServices); 

parameters: ppusServices, On return, this is the address of a newly-created 
lUPNPServices collection that can enumerate the services exposed by the device. 
This must be released when no longer needed. If the device exposes no services, this 
method will return a collection object with a length of zero. 

return values: S_OK, pusServices contains a list of the device's children. 

[propget, id(DISPID_UPNPDEVICE_SERVICEIDENTIFIER), 
helpstringC'returns the (optional) service identifier of the device")] 
HRESULT Serviceldentifier([out, retval] BSTR * pbstrServicelD ); 

parameters: pbstrServicelD, On return, this contains the address of a newly- 
allocated string containing the contents of the device's Serviceldentifier element, if the 
device specifies one. This value must be freed when no longer needed. If the device 
does not specify a Serviceldentifier value, this parameter will be set to null. 

return value: S_OK, bstrServicelD contains the service identifier of the device. 
pbstrServicelD must be freed. S_FALSE, the device did not specify a service identifier. 
*pbstrServicelD is set to null. 

note having a Serviceldentifier is mutually exclusive with having services. Any 
device will either have a list of services or a Serviceldentifier, but not both. 

[id(DISPID_UPNPDEVICEDESCRIPTION_LOADSMALLICON), 
helpstringC'loads a small (titlebar-sized) icon representing the device, encoded in the 
specified format")] 

HRESULT LoadSmalllcon([in] BSTR bstrEncodingFomiat, 
[out, retval] BSTR * pbstrlconURL); 
parameters: 
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bstrEncodingFormat, A string containing the mime-type representing the desired 
encoding format of the icon. pbstriconURL, On return, *pbstrlconURL contains a 
newly-allocated string representing the URL from which the icon can be loaded. 
This string must be freed when no longer needed. 

return values: S_OK, *pbstrlconURL contains a reference to an icon, 
encoded In the desired encoding format. 

[id(DISPID_UPNPDEVICEDESCRIPTION_LOADICON), 

helpstringC'loads a standard-sized icon representing the device, encoded in the 
specified format")] 

HRESULT Loadlcon([in] BSTR bstrEncodingFormat, 

[out, retval] BSTR * pbstriconURL); 

parameters: bstrEncodingFormat, A string containing the mime-type 
representing the desired encoding format of the icon. pbstriconURL, On return, 
*pbstrlconURL contains a newly-allocated string representing the URL from which 
the icon can be loaded. This string must be freed when no longer needed. 

return values: S_OK, *pbstrlconURL contains a reference to an icon, 
encoded in the desired encoding format. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_PRESENTATIONURL), 
helpstring ("obtains a presentation URL to a web page that can control the 
device")] 

HRESULT PresentationURL([out, retval] BSTR * pbstrURL); 

parameters: pbstrURL, on return, the address of a newly-allocated string 
containing the web-page-based control URL. If the device did not specify a 
presentation URL, an empy string ("") will be returned. 

return values:S_OK, bstrURL contains a newly-allocated URL that must be 
freed when no longer needed. S_FALSE, the device does not have a presentation 
URL. pbstrURL is set to null. 

[propget. id(DISPID_UPNPDEVICEDESCRIPTION_PHYSICALLOCATION), 
helpstringC'a set of properties describing the device's physical location")] 
HRESULT PhysicalLocation([out, retval] lUPNPPropertyBag * pupl); 
parameters: pupl on return, the address of a newly-allocated 
UPNPPropertyBag object which contains information about the device's physical 
location 

return values 
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S_OK upl contains a newly-allocated object that the caller must free when it 
is no longer needed. 

note: if the object does not provide any description information, an empy 
property bag will be returned. See SetPhysicalLocation for a listing of defined 
values in a physical location property bag. 

[propget, 

id(DISPID_UPNPDEVICEDESCRIPTION_CANSETPHYSICALLOCATION). 

helpstringfdenotes whether the physical location information of this device can 
be set")] 

HRESULT CanSetPhysicalLocation([out, retval] VARIANT_BOOL * pvarb); 
parameters: pvarb the address of a VARIANT_BOOL. This is true (1=0) on 
return when the device's physical location can be set (via SetPhysicalLocation) 
return values: S_OK varb is set to the appropriate value 

[id(DISPID_UPNPDEVICEDESCRIPTION_SETPHYSICALLOCATION). 
helpstring("writes a set of properties describing the device's physical location to 
the device")] 

HRESULT SetPhysicalLocation([in] lUPNPPropertyBag * pupl); 

parameters: pupl A UPNPPropertyBag object which contains the name- 
value pairs representing the device's current location, the function will not free the 
object. 

return values: S_OK he device has been updated with the supplied 
physical location information 

note: the following are standard values in the physical location property bag: 
country, campus, building, floor, wing, room, latitude, longitude, altitude. These 
values can be used programmatically to implement sorting or filtering functionality 
based on the device's location. Additionally the property bag supports the following 
value: description, which contains a user-displayable string representing a device's 
location which does not have programattic significance. Additionally, the physical 
location update will be made on the device alone, and will not be reflected in the 
current device object. After a successful call to this method, PhysicalLocation will 
continue to return the 'old' value. To read the device's current name, the caller 
must re-load the device's description. 

} 
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[propget, icl(DISPID_UPNPDEVICEDESCRIPTION_PRODUCTNAME), 
helpstringC'a displayable string containing the product name")] 
HRESULT ProductName([out. retval] BSTR * pbstr); 

parameters: pbstr on return, the address of a newly-allocated string 
containing the product name of the device. 

return values: S_OK pbstr contains a newly-allocated string that must 
be freed when no longer needed. 

[propget. id(DISPID_UPNPDEVICEDESCRIPTION_DESCRIPTION). 
helpstringC'displayable summary of the device's function")] 
HRESULT Description([out, retval] BSTR * pbstr); 

parameters: pbstr on return, the address of a newly-allocated string 
containing a short description of the device meaningful to the user. 

return values: S_OK pbstr contains a newly-allocated string that must 
be freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_MODELNAME). 
helpstringC'displayable model name")] 
HRESULT ModelName([out, retval] BSTR * pbstr); 

parameters: pbstr on return, the address of a newly-allocated string 
containing the manufactuer's model name of the device. 

return values: S_OK pbstr contains a newly-allocated string that must 
be freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_SERIALNUMBER). 
helpstringC'displayable serial number")] 
HRESULT SerialNumber([out, retval] BSTR * pbstr); 

parameters: pbstr on return, the address of a newly-allocated string 
containing the manufacturer's serial number of the device. 

return values: S_OK pbstr contains a newly-allocated string that must 
be freed when no longer needed. 

note: a device's serial number is not guaranteed to be globally unique. The 
DeviceUniqueName should always be used to distinguish devices. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_MANUFACTURERNAME), 
helpstringC'displayable manufacturer name")] 
HRESULT ManufacturerName([out, retval] BSTR * pbstr); 
parameters 



\ 



FIG. 38 

/ 

pbstr, on return, the address of a newly-allocated string containing the name of the 
device's manufactuer. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_MANUFACTURERURL), 
helpstringC'URL to the manufacturer's website")] 
HRESULT ManufacturerURL([out, retval] BSTR * pbstr); 

parameters: pbstr, on retum, the address of a newly-allocated string 
containing the URL of the manufacturer's website. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_MODELNAME), 
helpstringC'displayable model name")] 
HRESULT ModelName([out, retval] BSTR * pbstr); 

parameters: pbstr, on return, the address of a newly-allocated string 
containing the manufactuer's model name for the device. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_SUPPORTLIST), 
helpstringC'technical support contact information")] 
HRESULT SupportList([out, retval] BSTR * pbstr); 

parameters: pbstr, on retum, the address of a newly-allocated, multi-line 
string containing phone numbers and other information that can guide the user to 
technical support. This string must be freed when no longer needed. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 

[propget, id(DISPID_UPNPDEVICEDESCRIPTION_FAQLIST), 
helpstringfFAQ access display information")] 
HRESULT FAQList([out, retval] BSTR * pbstr); 

parameters: pbstr, on return, the address of a newly-allocated, multi-line 
string containing FAQ information that can provide the user with URLs at which 
device FAQs may be located. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 
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[propget, icl(DISPID_UPNPDEVICEDESCRIPTION_UPDATELIST), 

helpstringC'information explaining where the user can update the device's 
firmware")] 

HRESULT UpdateList([out, retval] BSTR * pbstr); 

parameters: pbstr, on return, the address of a newly-allocated, multi-line 
string containing Information and URLs from which the user can download updates 
for the device's firmware. 

return values: S_OK, pbstr contains a newly-allocated string that must be 
freed when no longer needed. 
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[ 

object, 

uuid(FDBC0C73-BDA3-4C66-AC4F-F2D96FDAD68C), 
dual, 

helpstringC'lUPNPDevices Interface"). 
pointer_default(unique) 

1 

lUPNPPropertyBag 
{ 

[propget, jd(DISPID_UPNP_PROPERTYBAG_READ), 
helpstringC'reads a value from the property bag")] 

HRESULT Read([in] BSTR bstrName, [out, retval] VARIANT * pvarResult); 
parameters: bstrName, name of the property to read, case is ignored. 
pvarResultvalue of the property, if the property doese not exist, this is of type 
VT_EMPTY 

return values: S_OK, the value was found in the property bag, and returned 
in pvarResult. S_FALSE, there was no value with the given name in the property 
bag. *pvarResult is of type \/T_EMPTY 

[propget, id(DISPID_UPNP_PROPERTYBAG_WRITE), 
helpstringC'writes a value to the property bag")] 
HRESULT Write([in] BSTR bstrName, [in] VARIANT * pvarValue); 

parameters: bstrName, name of the property to write, case is preserved 
when writing. The supplied value will replace any other values of the same name, 
even if they differ in case. pvarValue, value of the property to write. 

return values: S_OK, the value was written to the property bag, replacing the 
value currently associated with this property, if it existed. 

[propget, id(DISPlD_UPNP_PROPERTYBAG_DELETE). 
helpstring("removes a value from the property bag")] 
HRESULT Delete([in] BSTR bstrName); 

parameters: bstrName, name of the value to remove from the property gab. 
case is ignored when finding a value to remove. 

return values: S_OK, the value has been removed from the property bag. 
S_FALSE, the value was not found in the property bag. 

}: 
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[ 

object, 

uuid(A29501 9C-DC65-47DD-90DC-7FE91 8A1 AB44), 
dual, 

helpstringC'lUPNPService Interface"), 
pointer_default(unlque) 

] 

interface lUPNPService : IDispatch 
{ 

[id(1), helpstringC'method GetProperty")] 
HRESULT GetProperty( 
[in] BSTR bstrPropertyName, 
[out. retval] VARIANT *pValue 

); 

[id(2), helpstringC'method InvokeAction")] 
HRESULT lnvokeAction( 
[in] BSTR bstrActionName, 
[in] VARIANT saActionArgs, 
[out, retval] long *plStatus 

); 

[propget. id(3), helpstring("property DCPI")] 
HRESULT DCPI( 
[out, retval] BSTR *pVal 

): 

[propget, id(4), 

helpstringC'returns a manufactuer-defined extension property")] 
HRESULT VendorExtension([out. retval] VARIANT * pvarValue ); 

parameters: pvarValueOn return, this variant is filled with the value of the 
"extension" element. If none exists, pvarValue is set to VT_EMPTY 

return values: S_OK, varValue is set to the extension element. S_FALSE, 
no vendor extension element exists. pvarValue is VT_EMPTY 
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[ 

object, 

uuid(FDBC0C73-BDA3-4C66-AC4F-F2D96FDAD68C), 
dual, 

helpstringC'lUPNPDevices Interface"), 
pointer_default(unique) 

] 

interface lUPNPDevices : IDispatch 

{ 

[propget, id(1), helpstring("property Count")] 
HRESULT Count( 
[out, retval] long *pVal 

): 

[propget, id(DISPID_NEWENUM), helpstring("property _NewEnum")] 

HRESULT _NewEnum( 

[out, retval] LPUNKNOWN *pVal 

); 

[propget. id(DISPID_VALUE), helpstring("property Item")] 

HRESULT ltem( 

[in] long llndex, 

[out, retval] VARIANT *pVal 

): 
}: 
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[ 

object, 

uuid(3F8C8E9E-9A7A-4DC8-BC41 -FF31 FA374956), 
dual, 

helpstringC'lUPNPServices Interface"). 
pointer_default(unique) 

1 

interface lUPNPServices : IDispatch 
{ 

[propget, id(1), helpstring("property Count")] 
HRESULT Count( 
[out, retval] long *pVal 

): 

[propget, id(DISPID_NEWENUM), helpstring("property _NewEnum")] 

HRESULT _NewEnum( 

[out, retval] LPUNKNOWN *pVal 

); 

[propget, id(DISPID_VALUE), helpstring("property Item")] 

HRESULT ltem( 

[in] long llndex, 

[out, retval] VARIANT *pVal 

); 
}; 
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<contract> 

<protocol id="protocolDef' > 
<HTTP version="1.1"> 

<URL> http://investor.msn.com/stockquote </URL> 
<M-POST> 

<MAN> http://www.upnp.org/service-control/m-post </MAN> 
<IVI-POST> 

<HEADER name="Content-Type" value="text/xml" /> 
</HTTP> 
</protocol> 

<RequestResponse name-'getQuote"> 

<protocol is="protocolDef ' /> 

<in is="symbol" /> 

<out is="stockQuote" /> 

<error is="error" /> 
</RequestResponse> 

<RequestResponse name="getQuotes"> 

<protocol is="protocolDef' /> 

<in is="symbols" /> 

<out is="stockQuotes" /> 

<error is="error" /> 
</Req uestResponse> 

<!-- // schema definition follows ~> 

<schema xmlns="urn:schema-microsoft-com:xml-data" 
xmlns:dt="urn:schema-microsoft-com:datatypes"> 

<ElementType name="symboi" dt:type="string" /> 

<ElementType name="symbols"> 
<element type="symbol" maxOccurs-'*" /> 
</ElementType> 

<ElementType name="stockQuote"> 
<element type="company" /> 
<element type="ticker" /> 
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<element type="previousClose" /> 
<element type="openingTrade" /> 
<element type="lastTrade" /> 
<element type="volume" /> 
</ElementType> 

<ElementType dt:type="string" name="company" /> 
<ElementType dt:type="string" name="ticker" /> 
<ElementType dt:type="string" name="previousClose" /> 
<ElementType dt:type="string" name="openingTrade" /> 
<ElementType dt:type="string" name="lastTrade" /> 
<ElementType dt:type="string" name="volume" /> 

<ElementType name="stockQuotes"> 
<element name="stockQuote" maxOccurs="*" /> 
</Element> 

<ElementType name="error"> 
<element type="reason" /> 
</ElementType> 

<ElementType dt:type="string" nanne="reason" /> 
</schema> 
</contract> 

Request for "getQuote" 

M-POST /stockquotes HTTP/1.1 
Host: amarg5:8586 
Content-Type: text/xml 

Man: "http://wvvw.upnp.org/service-control/iTi-post"; ns=01 
01-MethodName: getQuotes 
01-MessageType: Call 
Accept-Language: en-gb, en;q=0.8 

Referer: http://amarg5/uPnPService/Services/Stock/Client/ticker.htm 
Content-Length: 327 

User-Agent: Mozilla/4.0 (compatible; MSIE 5.01; Windows NT 5.0) 
Connection: Keep-Alive 
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<symbol>MSFT</symbol> 
Response for "getQuote" 

HTTP/1.1 200 OK 

Connection: close 

Cache-Control: private 

Date: Mon Aug 16 15:37:35 PDT 1999 

Expires: Mon Aug 16 15:37:35 PDT 1999 

Content-Type: text/xml 

Content-Length: 7912 

Man: "http://www.upnp.org/service-control/m-post"; ns=01 
Ext: 

01-MessageType: CallResponse 

<stockQuote> 
<company>Microsoft%20Corporation</company> 
<ticker>MSFT</ticker> 

<previousClose>84%2011/16</previousClose> 
<openingTrade>85%201/16</openlngTrade> 
<lastTrade>84%205/16</lastTrade> 
<volume>28.66%20Mil</volume> 
</stockQuote> 
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<!- XDR Schema for protocol section of contract -> 

<schema name="contract" 

xmlns="urn:schema-microsoft-com:xml-data" 
xmlns:dt="urn:schema-microsoft-com:datatypes"> 

<ElementType name-'contract" 

xmlns:protocolNS="contract-protocol" 
xmlns:msgPatternNS="contract-msgPatterns" 
xmlns:schemaNS="urn:schema-microsoft-com:xml-data"> 

<element type-'protocolNSiprotocol" /> 

<element type="msgPatternNS:RequestResponse" minOccurs="0" 
maxOccurs-'*" /> 

<element type="msgPatternNS:SolicitResponse" minOccurs="0" maxOccurs="* 

/> 

<element type="schemaNS:schema" minOccurs="0" maxOccurs="*" /> 

</ElementType> 
</schema> 
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Protocol 

<!- XDR Schema for protocol section of contract ~> 

<schema name="contract-protocor' 

xmlns="urn:schenfia-microsoft-com:xml-data" 
xmlns:clt="urn:schema-microsoft-com:datatypes"> 

<ElementType name="protocor> 

<!_ ID -> 

<AttributeType name="id" dt:type="id" /> 
<Attribute type="id" /> 

<group order="one"> 

<element xmlns:http="contract-protocol-HTTP" type="http:HTTP" /> 
<element xmlns:gena="contract-protocol-GENA" type="gena:GENA" /> 
// other protocol definitions go here 
</group> 

</ElementType> 

</schema> 
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HTTP 

<!- XDR Schema for HTTP section of contract -> 

<schema nanne="contract-protocol-HTTP" 

xmlns="urn:schema-microsoft-com:xml-clata" 
xmlns:dt="urn:schema-microsoft-com:datatypes"> 

<ElementType name="HTTP"> 

<!- HTTP version ~> 

<AttributeType name="VERSION" dt:type="string" default= 
<Attribute type='VERSION" /> 

<!-- The Verb to use -> 
<group order="one"> 

<element type="GEr' /> 

<element type="POSr' /> 

<element type="M-POST" /> 
</group> 

<!- The protocol data -> 

<element type="URL" /> 

<element type="QUERY" minOccurs="0" /> 

<element type="HEADER" minOccurs="0" /> 

</ElementType> 

<ElementType name="URL" dt:type="string" /> 

<ElementType name="QUERY"> 

<attribute type="name" /> 

<attribute type="value" /> 

<attribute type="required" /> 
</ElementType> 
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<ElementType name="HEADER"> 

ottribute type-'name" /> 

<attribute type="value" required="yes" /> 
</ElementType> 

<!- Verb declarations -> 
<ElementType name="GErV> 

<ElementType name="POST"> 
<element type="PARAM" minOccurs="0" maxOccurs="*" /> 
</ElementType> 

<ElementType name="PARAM"> 

<element type="name" /> 

<element type-'default" /> 

<element type="value" /> 

<element type="required" /> 
</ElementType> 

<AttributeType name="name" dt:type="string" required="yes" /> 
<AttributeType name="default" dt:type="string" /> 
<AttributeType name="value" dt:type="string" /> 
<AttributeType name="required" dt:type="boolean" default="no" /> 

<ElementType name="M-POST"> 
olement type="MAN" /> 
</ElementType> 

<ElementType name="MAN" dt:type="string" /> 
</schema> 
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